/** * Licensed to the Apache Software Foundation (ASF) under one * or more contributor license agreements. See the NOTICE file * distributed with this work for additional information * regarding copyright ownership. The ASF licenses this file * to you under the Apache License, Version 2.0 (the * "License"); you may not use this file except in compliance * with the License. You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, * software distributed under the License is distributed on an * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY * KIND, either express or implied. See the License for the * specific language governing permissions and limitations * under the License. */ package org.apache.lens.cli.commands; import java.util.Iterator; import java.util.List; import javax.ws.rs.ProcessingException; import org.apache.lens.api.APIResult; import org.apache.lens.api.LensSessionHandle; import org.apache.lens.cli.commands.annotations.UserDocumentation; import org.apache.lens.cli.config.LensCliConfigConstants; import org.apache.lens.client.LensClient; import org.apache.lens.client.LensClientConfig; import org.slf4j.LoggerFactory; import org.springframework.shell.core.ExitShellRequest; import org.springframework.shell.core.annotation.CliCommand; import org.springframework.shell.core.annotation.CliOption; import org.springframework.stereotype.Component; import ch.qos.logback.classic.Level; import ch.qos.logback.classic.Logger; import ch.qos.logback.classic.LoggerContext; import ch.qos.logback.classic.PatternLayout; import ch.qos.logback.classic.spi.ILoggingEvent; import ch.qos.logback.core.Appender; import ch.qos.logback.core.ConsoleAppender; import ch.qos.logback.core.Context; import com.google.common.base.Joiner; import lombok.NonNull; import lombok.extern.slf4j.Slf4j; /** * The Class LensConnectionCommands. */ @Slf4j @Component @UserDocumentation(title = "Commands for Session Management", description = "Opening the lens CLI shell is equivalent to open a session with lens server." + "This section provides all the commands available for in shell which are applicable for the full session.") public class LensConnectionCommands extends BaseLensCommand { /** * Sets the param. * * @param keyval the keyval * @return the string */ @CliCommand(value = "set", help = "Assign <value> to session parameter specified with <key> on lens server") public String setParam(@CliOption(key = {""}, mandatory = true, help = "<key>=<value>") String keyval) { String[] pair = keyval.split("="); if (pair.length != 2) { return "Error: Pass parameter as <key>=<value>"; } if (pair[0].startsWith(LensCliConfigConstants.LENS_CLI_PREFIX) || pair[0].startsWith(LensClientConfig.CLIENT_PFX)) { getClient().getConf().set(pair[0], pair[1]); return "Client side Set " + pair[0] + "=" + pair[1]; } else { APIResult result = getClient().setConnectionParam(pair[0], pair[1]); return result.getMessage(); } } /** * Gets the param. * * @param param the param * @return the param */ @CliCommand(value = "get", help = "Fetches and prints session parameter specified with name <key> from lens server") public String getParam(@CliOption(key = {"", "key"}, mandatory = true, help = "<key>") String param) { return Joiner.on("\n").skipNulls().join(getClient().getConnectionParam(param)); } /** * Show parameters. * * @return the string */ @CliCommand(value = "show params", help = "Fetches and prints all session parameter from lens server") public String showParameters() { List<String> params = getClient().getConnectionParam(); return Joiner.on("\n").skipNulls().join(params); } /** * Adds the jar. * * @param path the path * @return the string */ @CliCommand(value = "add jar", help = "Adds jar resource to the session") public String addJar( @CliOption(key = {"", "path"}, mandatory = true, help = "<path-to-jar-on-server-side>") @NonNull String path) { APIResult result = getClient().addJarResource(path); return result.getMessage(); } /** * Removes the jar. * * @param path the path * @return the string */ @CliCommand(value = "remove jar", help = "Removes a jar resource from session") public String removeJar( @CliOption(key = {"", "path"}, mandatory = true, help = "<path-to-jar-on-server-side>") @NonNull String path) { APIResult result = getClient().removeJarResource(path); return result.getMessage(); } /** * Adds the file. * * @param path the path * @return the string */ @CliCommand(value = "add file", help = "Adds a file resource to session") public String addFile( @CliOption(key = {"", "path"}, mandatory = true, help = "<path-to-file-on-server-side>") @NonNull String path) { APIResult result = getClient().addFileResource(path); return result.getMessage(); } /** * Removes the file. * * @param path the path * @return the stringadd */ @CliCommand(value = "remove file", help = "removes a file resource from session") public String removeFile( @CliOption(key = {"", "path"}, mandatory = true, help = "<path-to-file-on-server-side>") @NonNull String path) { APIResult result = getClient().removeFileResource(path); return result.getMessage(); } /** * List resources. * * @return the string */ @CliCommand(value = "list resources", help = "list all resources from session. If type is provided, " + " lists resources of type <resource-type>. Valid values for type are jar and file.") public String listResources(@CliOption(key = {"", "type"}, mandatory = false, help = "<resource-type>") String type) { List<String> resources = getClient().listResources(type); if (resources == null) { return "No resources found"; } return Joiner.on("\n").skipNulls().join(resources); } /** * Get the current session handle */ @CliCommand(value = {"session"}, help = "Print the current session handle") public String getSessionHandle() { LensSessionHandle sessionHandle = getClient().getConnection().getSessionHandle(); if (sessionHandle != null) { return "Session Handle: " + sessionHandle.getPublicId(); } else { return "Session not established"; } } /** * Enables to show all class level logs on cli * @param enable */ @CliCommand(value = {"debug"}, help = "prints all class level logs and verbose logs on cli for debugging purpose." + " 'debug false' to turn off all class level logging and verbose level logging ") public void debug(@CliOption(key = {"", "enable"}, mandatory = false, unspecifiedDefaultValue = "true") boolean enable) { Logger logger = LoggerUtil.getRootLogger(); Logger cliLogger = LoggerUtil.getCliLogger(); if (enable) { LoggerUtil.addConsoleAppenderIfNotPresent(logger); LoggerUtil.addConsoleAppenderIfNotPresent(cliLogger); } else { LoggerUtil.removeConsoleAppender(logger); LoggerUtil.removeConsoleAppender(cliLogger); } } /** * Enables to show only cliLogger(verbose) logs on cli * @param enable */ @CliCommand(value = {"verbose"}, help = "Show cliLogger logs on cli. 'verbose false' turns off the cliLogger logs on console") public void verbose(@CliOption(key = {"", "enable"}, mandatory = false, unspecifiedDefaultValue = "true") boolean enable) { Logger cliLogger = LoggerUtil.getCliLogger(); if (enable) { LoggerUtil.addConsoleAppenderIfNotPresent(cliLogger); } else { LoggerUtil.removeConsoleAppender(cliLogger); } } private static class LoggerUtil { private static Logger logger; private static Logger cliLogger; public static Logger getRootLogger() { if (logger == null) { logger = (Logger) LoggerFactory.getLogger(Logger.ROOT_LOGGER_NAME); } return logger; } public static Logger getCliLogger() { if (cliLogger == null) { cliLogger = (Logger) LoggerFactory.getLogger(LensClient.CLILOGGER); } return cliLogger; } public static ConsoleAppender<ILoggingEvent> getConsoleAppender(PatternLayout layout, Context context) { ConsoleAppender<ILoggingEvent> consoleAppender = new ConsoleAppender<ILoggingEvent>(); consoleAppender.setContext(context); consoleAppender.setLayout(layout); consoleAppender.start(); return consoleAppender; } public static PatternLayout getPatternLayout(LoggerContext context) { String conversionPattern = "%d [%t] %F %-7p - %m%n"; return getPatternLayout(conversionPattern, context); } public static PatternLayout getPatternLayout(String conversionPattern, Context context) { PatternLayout layout = new PatternLayout(); layout.setPattern(conversionPattern); layout.setContext(context); layout.start(); return layout; } /** * Check whether ConsoleAppender was already added * @param logger * @return true if it was already added * false if it does not contain ConsoleAppender */ public static void addConsoleAppenderIfNotPresent(Logger logger) { boolean isConsoleAppenderAdded = false; PatternLayout layout = null; Iterator<Appender<ILoggingEvent>> appenderSeries = logger.iteratorForAppenders(); while (appenderSeries.hasNext()) { Appender<ILoggingEvent> appender = (Appender<ILoggingEvent>) appenderSeries.next(); if (appender instanceof ConsoleAppender) { isConsoleAppenderAdded = true; break; } } if (!isConsoleAppenderAdded) { if (layout == null) { layout = LoggerUtil.getPatternLayout(logger.getLoggerContext()); } if (logger.getLevel() == null) { logger.setLevel(Level.DEBUG); } ConsoleAppender<ILoggingEvent> consoleAppender = LoggerUtil.getConsoleAppender(layout, logger.getLoggerContext()); logger.addAppender(consoleAppender); } } public static void removeConsoleAppender(Logger logger) { Iterator<Appender<ILoggingEvent>> appenderSeries = logger.iteratorForAppenders(); while (appenderSeries.hasNext()) { Appender<ILoggingEvent> appender = (Appender<ILoggingEvent>) appenderSeries.next(); if (appender instanceof ConsoleAppender) { logger.detachAppender(appender); } } } } /** * Quit shell. * * @return the exit shell request */ @CliCommand(value = {"close", "bye"}, help = "Releases all resources of the server session and exits the shell") public ExitShellRequest quitShell() { try { closeClientConnection(); return ExitShellRequest.NORMAL_EXIT; } catch (ProcessingException e) { log.error("Error while closing client connection.", e); return ExitShellRequest.FATAL_EXIT; } } }